1 /* 2 * Copyright 2002-2013 the original author or authors. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package org.springframework.beans.factory.config; 18 19 import org.springframework.beans.factory.BeanDefinitionStoreException; 20 import org.springframework.beans.factory.BeanFactory; 21 import org.springframework.beans.factory.BeanFactoryAware; 22 import org.springframework.beans.factory.BeanNameAware; 23 import org.springframework.util.StringValueResolver; 24 25 /** 26 * Abstract base class for property resource configurers that resolve placeholders 27 * in bean definition property values. Implementations <em>pull</em> values from a 28 * properties file or other {@linkplain org.springframework.core.env.PropertySource 29 * property source} into bean definitions. 30 * 31 * <p>The default placeholder syntax follows the Ant / Log4J / JSP EL style: 32 * 33 *<pre class="code">${...}</pre> 34 * 35 * Example XML bean definition: 36 * 37 *<pre class="code">{@code 38 *<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"/> 39 * <property name="driverClassName" value="}${driver}{@code "/> 40 * <property name="url" value="jdbc:}${dbname}{@code "/> 41 *</bean> 42 *}</pre> 43 * 44 * Example properties file: 45 * 46 * <pre class="code"> driver=com.mysql.jdbc.Driver 47 * dbname=mysql:mydb</pre> 48 * 49 * Annotated bean definitions may take advantage of property replacement using 50 * the {@link org.springframework.beans.factory.annotation.Value @Value} annotation: 51 * 52 *<pre class="code">@Value("${person.age}")</pre> 53 * 54 * Implementations check simple property values, lists, maps, props, and bean names 55 * in bean references. Furthermore, placeholder values can also cross-reference 56 * other placeholders, like: 57 * 58 *<pre class="code">rootPath=myrootdir 59 *subPath=${rootPath}/subdir</pre> 60 * 61 * In contrast to {@link PropertyOverrideConfigurer}, subclasses of this type allow 62 * filling in of explicit placeholders in bean definitions. 63 * 64 * <p>If a configurer cannot resolve a placeholder, a {@link BeanDefinitionStoreException} 65 * will be thrown. If you want to check against multiple properties files, specify multiple 66 * resources via the {@link #setLocations locations} property. You can also define multiple 67 * configurers, each with its <em>own</em> placeholder syntax. Use {@link 68 * #ignoreUnresolvablePlaceholders} to intentionally suppress throwing an exception if a 69 * placeholder cannot be resolved. 70 * 71 * <p>Default property values can be defined globally for each configurer instance 72 * via the {@link #setProperties properties} property, or on a property-by-property basis 73 * using the default value separator which is {@code ":"} by default and 74 * customizable via {@link #setValueSeparator(String)}. 75 * 76 * <p>Example XML property with default value: 77 * 78 *<pre class="code">{@code 79 * <property name="url" value="jdbc:}${dbname:defaultdb}{@code "/> 80 *}</pre> 81 * 82 * @author Chris Beams 83 * @author Juergen Hoeller 84 * @since 3.1 85 * @see PropertyPlaceholderConfigurer 86 * @see org.springframework.context.support.PropertySourcesPlaceholderConfigurer 87 */ 88 public abstract class PlaceholderConfigurerSupport extends PropertyResourceConfigurer 89 implements BeanNameAware, BeanFactoryAware { 90 91 /** Default placeholder prefix: {@value} */ 92 public static final String DEFAULT_PLACEHOLDER_PREFIX = "${"; 93 94 /** Default placeholder suffix: {@value} */ 95 public static final String DEFAULT_PLACEHOLDER_SUFFIX = "}"; 96 97 /** Default value separator: {@value} */ 98 public static final String DEFAULT_VALUE_SEPARATOR = ":"; 99 100 101 /** Defaults to {@value #DEFAULT_PLACEHOLDER_PREFIX} */ 102 protected String placeholderPrefix = DEFAULT_PLACEHOLDER_PREFIX; 103 104 /** Defaults to {@value #DEFAULT_PLACEHOLDER_SUFFIX} */ 105 protected String placeholderSuffix = DEFAULT_PLACEHOLDER_SUFFIX; 106 107 /** Defaults to {@value #DEFAULT_VALUE_SEPARATOR} */ 108 protected String valueSeparator = DEFAULT_VALUE_SEPARATOR; 109 110 protected boolean ignoreUnresolvablePlaceholders = false; 111 112 protected String nullValue; 113 114 private BeanFactory beanFactory; 115 116 private String beanName; 117 118 119 /** 120 * Set the prefix that a placeholder string starts with. 121 * The default is {@value #DEFAULT_PLACEHOLDER_PREFIX}. 122 */ 123 public void setPlaceholderPrefix(String placeholderPrefix) { 124 this.placeholderPrefix = placeholderPrefix; 125 } 126 127 /** 128 * Set the suffix that a placeholder string ends with. 129 * The default is {@value #DEFAULT_PLACEHOLDER_SUFFIX}. 130 */ 131 public void setPlaceholderSuffix(String placeholderSuffix) { 132 this.placeholderSuffix = placeholderSuffix; 133 } 134 135 /** 136 * Specify the separating character between the placeholder variable 137 * and the associated default value, or {@code null} if no such 138 * special character should be processed as a value separator. 139 * The default is {@value #DEFAULT_VALUE_SEPARATOR}. 140 */ 141 public void setValueSeparator(String valueSeparator) { 142 this.valueSeparator = valueSeparator; 143 } 144 145 /** 146 * Set a value that should be treated as {@code null} when 147 * resolved as a placeholder value: e.g. "" (empty String) or "null". 148 * <p>Note that this will only apply to full property values, 149 * not to parts of concatenated values. 150 * <p>By default, no such null value is defined. This means that 151 * there is no way to express {@code null} as a property 152 * value unless you explicitly map a corresponding value here. 153 */ 154 public void setNullValue(String nullValue) { 155 this.nullValue = nullValue; 156 } 157 158 /** 159 * Set whether to ignore unresolvable placeholders. 160 * <p>Default is "false": An exception will be thrown if a placeholder fails 161 * to resolve. Switch this flag to "true" in order to preserve the placeholder 162 * String as-is in such a case, leaving it up to other placeholder configurers 163 * to resolve it. 164 */ 165 public void setIgnoreUnresolvablePlaceholders(boolean ignoreUnresolvablePlaceholders) { 166 this.ignoreUnresolvablePlaceholders = ignoreUnresolvablePlaceholders; 167 } 168 169 /** 170 * Only necessary to check that we're not parsing our own bean definition, 171 * to avoid failing on unresolvable placeholders in properties file locations. 172 * The latter case can happen with placeholders for system properties in 173 * resource locations. 174 * @see #setLocations 175 * @see org.springframework.core.io.ResourceEditor 176 */ 177 @Override 178 public void setBeanName(String beanName) { 179 this.beanName = beanName; 180 } 181 182 /** 183 * Only necessary to check that we're not parsing our own bean definition, 184 * to avoid failing on unresolvable placeholders in properties file locations. 185 * The latter case can happen with placeholders for system properties in 186 * resource locations. 187 * @see #setLocations 188 * @see org.springframework.core.io.ResourceEditor 189 */ 190 @Override 191 public void setBeanFactory(BeanFactory beanFactory) { 192 this.beanFactory = beanFactory; 193 } 194 195 196 protected void doProcessProperties(ConfigurableListableBeanFactory beanFactoryToProcess, 197 StringValueResolver valueResolver) { 198 199 BeanDefinitionVisitor visitor = new BeanDefinitionVisitor(valueResolver); 200 201 String[] beanNames = beanFactoryToProcess.getBeanDefinitionNames(); 202 for (String curName : beanNames) { 203 // Check that we're not parsing our own bean definition, 204 // to avoid failing on unresolvable placeholders in properties file locations. 205 if (!(curName.equals(this.beanName) && beanFactoryToProcess.equals(this.beanFactory))) { 206 BeanDefinition bd = beanFactoryToProcess.getBeanDefinition(curName); 207 try { 208 visitor.visitBeanDefinition(bd); 209 } 210 catch (Exception ex) { 211 throw new BeanDefinitionStoreException(bd.getResourceDescription(), curName, ex.getMessage(), ex); 212 } 213 } 214 } 215 216 // New in Spring 2.5: resolve placeholders in alias target names and aliases as well. 217 beanFactoryToProcess.resolveAliases(valueResolver); 218 219 // New in Spring 3.0: resolve placeholders in embedded values such as annotation attributes. 220 beanFactoryToProcess.addEmbeddedValueResolver(valueResolver); 221 } 222 223 }